package com.lzh.java1;

/**
 * @author liuzhihao
 * @create 2021-02-18 14:25
 */

import org.junit.Test;

import java.lang.annotation.Annotation;
import java.util.ArrayList;

/**
 * 1.理解Annotation
 * ①jdk 5.0 新增的功能
 *
 * ②Annotation 其实就是代码里的特殊标记, 这些标记可以在编译, 类加
 * 载, 运行时被读取, 并执行相应的处理。通过使用 Annotation, 程序员
 * 可以在不改变原有逻辑的情况下, 在源文件中嵌入一些补充信息。代
 * 码分析工具、开发工具和部署工具可以通过这些补充信息进行验证
 * 或者进行部署。
 *
 * ③在JavaEE/Android中注解占据了更重要的角色，例如
 * 用来配置应用程序的任何切面，代替JavaEE旧版中所遗留的繁冗
 * 代码和XML配置等。
 *
 * 2.Annotation的使用示例：
 * 示例一：生成文档相关的注解
 * 示例二：在编译时进行格式检查(JDK内置的三个基本注解)
 * @Override: 限定重写父类方法, 该注解只能用于方法
 * @Deprecated: 用于表示所修饰的元素(类, 方法等)已过时。通常是因为
 * 所修饰的结构危险或存在更好的选择
 * @SuppressWarnings: 抑制编译器警告
 *
 * 示例三：跟踪代码依赖性，实现替代配置文件功能
 *
 * 3.如何自定义注解：参照@SuppressWarnings定义
 *  ①注解声明为： @interface
 *  ②内部定义成员，通常使用value表示
 *  ③可以指定成员的默认值，使用default定义
 *  ④如果自定义注解没有成员，表明是一个标识作用
 *
 * 如果注解有成员，在使用注解时，需要指明成员的值
 *
 * 自定义注解必须配上注解的信息处理流程（使用反射）才有意义。
 *
 * 自定义注解通常会指明两个元注解：Retention,Target
 * 4.jdk 提供的元注解
 * 元注解：对现有的注解进行解释说明
 * Retention:指定所修饰的Annotation 的声明周期：SOURCE\CLASS（默认行为）\RUNTIME
 *          只有声明为RUNTIME声明周期的注解，才能通过反射获取。
 * Target:用于指定被修饰的Annotion能用于修饰哪些程序元素
 * *****出现频率较低********
 * Documented:表示所修饰的注解在被javadoc解析时，保留下来
 * Inherited:被它修饰的 Annotation将具有继承性
 *
 * 5.通过反射获取注解信息
 *
 * 6.jdk 8中注解的新特性：可重复注解、类型注解
 *
 * 6.1 可重复注解：①在MyAnnotation上声明@Repeeatable,成员值为MyAnnotation.class
 *               ②MyAnnotation的Target和Retention等元注解与MyAnnotations相同
 *
 * 6.2类型注解：
 *  ElementType.TYPE_PARAMETER 表示该注解能写在类型变量的声明语
 * 句中（如：泛型声明）。
 *  ElementType.TYPE_USE 表示该注解能写在使用类型的任何语句中。
 */

public class AnnotationTest {

    @Test
    public void testGetAnntation(){

        Class<Student> clazz = Student.class;
        Annotation[] annotation = clazz.getAnnotations();
        for (int i = 0; i < annotation.length; i++) {
            System.out.println(annotation[i]);
        }

    }
}

@MyAnnotation(value = "hi")
@MyAnnotation(value = "abc")
//@MyAnnotations({@MyAnnotation(value = "hi"), @MyAnnotation(value = "abc")})
class Person{

    private String name;
    private int age;
    @MyAnnotation()
    public Person() {
    }

    public Person(String name, int age) {
        this.name = name;
        this.age = age;
    }
    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }
    public void walk(){
        System.out.println("人走路");
    }
    public void eat(){
        System.out.println("人吃饭");
    }

}
interface  Info{
    void show();
}
class Student extends Person implements Info{

    @Override
    public void walk() {
        System.out.println("学生走路");
    }

    @Override
    public void show() {

    }
}

class Generic<@MyAnnotation T>{

    public void show() throws @MyAnnotation RuntimeException{
        ArrayList<@MyAnnotation String> list = new ArrayList<>();

        int num = (@MyAnnotation int) 10L;
    }

}

